JdbcTemplate简介

  • 为了使JDBC更加易于使用,Spring在JDBC API定义了一个抽象层,以此建立一个JDBC存取框架
  • 作为Spring JDBC框架的核心,JDBC模板的设计目的是为不同类型的JDBC操作提供模板方法,每个模板方法都能控制整个过程,并允许覆盖过程中的特定任务

JdbcTemplate的使用

  • 加入jar包

    • c3p0-0.9.1.2.jar
    • com.springsource.net.sf.cglib-2.2.0.jar
    • com.springsource.org.aopalliance-1.0.0.jar
    • com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
    • commons-logging-1.1.3.jar
    • mysql-connector-java-5.1.7-bin.jar
    • spring-aop-4.0.0.RELEASE.jar
    • spring-aspects-4.0.0.RELEASE.jar
    • spring-beans-4.0.0.RELEASE.jar
    • spring-context-4.0.0.RELEASE.jar
    • spring-core-4.0.0.RELEASE.jar
    • spring-expression-4.0.0.RELEASE.jar
    • spring-jdbc-4.0.0.RELEASE.jar
    • spring-orm-4.0.0.RELEASE.jar
    • spring-tx-4.0.0.RELEASE.jar
    • spring-web-4.0.0.RELEASE.jar
    • spring-webmvc-4.0.0.RELEASE.jar
  • 建立DB属性文件db.properties(将数据库配置信息放置在Spring配置文件之外)

    1
    2
    3
    4
    5
    6
    jdbc.user=xxxx
    jdbc.password=xxxx
    jdbc.driverClass=com.mysql.jdbc.Driver
    jdbc.jdbcUrl=jdbc:mysql:///xxxx
    jdbc.initPoolSize=x
    jdbc.maxPoolSize=x
  • 创建applicationContext.xml文件

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd">
    <!--指定数据库配置文件的存放位置-->
    <context:property-placeholder location="classpath:db.properties"/>
    <!--配置C3P0数据源-->
    <bean id="dataSource"
    class="com.mchange.v2.c3p0.ComboPooledDataSource">
    <!--EL表达式获取-->
    <property name="user" value="${jdbc.user}"></property>
    <property name="password" value="${jdbc.password}"></property>
    <property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
    <property name="driverClass" value="${jdbc.driverClass}"></property>
    <property name="initialPoolSize" value="${jdbc.initPoolSize}"></property>
    <property name="maxPoolSize" value="${jdbc.maxPoolSize}"></property>
    </bean>
    <!-- 配置Spring的JdbcTemplate -->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    <property name="dataSource" ref="dataSource"></property>
    </bean>
    </beans>
  • 建立实体类

    • Employee.java

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      35
      package com.glemontree.spring.jdbc;
      public class Employee {
      private Integer id;
      private String lastName;
      private String email;
      private Department department;
      public Integer getId() {
      return id;
      }
      public void setId(Integer id) {
      this.id = id;
      }
      public String getLastName() {
      return lastName;
      }
      public void setLastName(String lastName) {
      this.lastName = lastName;
      }
      public String getEmail() {
      return email;
      }
      public void setEmail(String email) {
      this.email = email;
      }
      public Department getDepartment() {
      return department;
      }
      public void setDepartment(Department department) {
      this.department = department;
      }
      @Override
      public String toString() {
      return "Employee [id=" + id + ", lastName=" + lastName + ", email="
      + email + ", department=" + department + "]";
    1
    }

    }

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    - Department.java
    ```java
    package com.glemontree.spring.jdbc;
    public class Department {
    private Integer id;
    private String name;
    public Integer getId() {
    return id;
    }
    public void setId(Integer id) {
    this.id = id;
    }
    public String getName() {
    return name;
    }
    public void setName(String name) {
    this.name = name;
    }
    @Override
    public String toString() {
    return "Department [id=" + id + ", name=" + name + "]";
    }
    }
  • 建立单元测试类

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    package com.glemontree.spring.jdbc;
    import java.nio.channels.SelectableChannel;
    import java.sql.SQLException;
    import java.util.ArrayList;
    import java.util.List;
    import javax.sql.DataSource;
    import javax.swing.text.DefaultEditorKit.InsertBreakAction;
    import org.junit.Test;
    import org.springframework.context.ApplicationContext;
    import org.springframework.context.support.ClassPathXmlApplicationContext;
    import org.springframework.jdbc.core.BeanPropertyRowMapper;
    import org.springframework.jdbc.core.JdbcTemplate;
    import org.springframework.jdbc.core.RowMapper;
    public class JdbcTest {
    private ApplicationContext ctx = null;
    private JdbcTemplate jdbcTemplate;
    {
    ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
    jdbcTemplate = (JdbcTemplate) ctx.getBean("jdbcTemplate");
    }
    @Test
    public void test() {
    DataSource dataSource = ctx.getBean(DataSource.class);
    try {
    System.out.println(dataSource.getConnection());
    } catch (SQLException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
    }
    }
    @Test
    public void testUpdate() {
    String sql = "UPDATE employees SET last_name = ? WHERE ID = ?";
    jdbcTemplate.update(sql, "Jack", 5);
    }
    /**
    * 执行批量更新:批量INSERT,UPDAET,DELETE
    * 最后一个参数是Object[]的List类型,因为修改一条记录需要一个Object[],那么多条就需要多个Object[],即List<Object[]>
    */
    @Test
    public void testBatchUpdate() {
    String sql = "INSERT INTO employees(last_name, email, dept_id) values(?,?,?)";
    List<Object[]> batchArgs = new ArrayList<Object[]>();
    batchArgs.add(new Object[]{"AA", "aa@gmail.com", 1});
    batchArgs.add(new Object[]{"BB", "bb@gmail.com", 1});
    batchArgs.add(new Object[]{"CC", "cc@gmail.com", 1});
    batchArgs.add(new Object[]{"DD", "dd@gmail.com", 1});
    batchArgs.add(new Object[]{"EE", "ee@gmail.com", 1});
    jdbcTemplate.batchUpdate(sql, batchArgs);
    }
    /**
    * 从数据库中获取一条记录,实际得到对应的一个对象
    * 注意:不是调用queryForObject(String sql, Class<Employee> requiredType, Object... args)方法
    * 而需要调用queryForObject(String sql, RowMapper<Employee> rowMapper, Object... args)
    * 其中的RowMapper指定如何映射结果集的行,常用的实现类为BeanPropertyRowMapper
    * 使用SQL中列的别名完成列名和类的属性名的映射,例如last_name lastName
    * 不支持级联属性,JdbcTemplate到底是一个JDBC的一个小工具,而不是ORM框架
    */
    @Test
    public void QueryForObject() {
    String sql = "SELECT id, last_name lastName, email, dept_id as \"department.id\" FROM employees WHERE id = ?";
    RowMapper<Employee> rowMapper = new BeanPropertyRowMapper<Employee>(Employee.class);
    Employee employee = jdbcTemplate.queryForObject(sql, rowMapper, 1);
    System.out.println(employee);
    }
    /**
    * 查找实体类的集合
    * 注意调用的不是queryForList()方法
    */
    @Test
    public void testQueryForList() {
    String sql = "SELECT id, last_name lastName, email FROM employees WHERE id > ?";
    RowMapper<Employee> rowMapper = new BeanPropertyRowMapper<Employee>(Employee.class);
    List<Employee> employees = jdbcTemplate.query(sql, rowMapper, 5);
    System.out.println(employees);
    }
    /*
    * 获取单个列的值,或做统计查询
    * 使用queryForObject(String sql, Class<Long> requiredType)方法
    */
    @Test
    public void testQueryForObject2() {
    String sql = "SELECT count(id) FROM employees";
    long count = jdbcTemplate.queryForObject(sql, Long.class);
    System.out.println(count);
    }
    }

    在单元测试类中进行不同情况的测试,使用JdbcTemplate。

  • 实际使用

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    package com.glemontree.spring.jdbc;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.jdbc.core.BeanPropertyRowMapper;
    import org.springframework.jdbc.core.JdbcTemplate;
    import org.springframework.jdbc.core.RowMapper;
    import org.springframework.stereotype.Repository;
    @Repository
    public class EmployeeDao {
    @Autowired
    private JdbcTemplate jdbcTemplate;
    public Employee get(Integer id) {
    String sql = "SELECT id, last_name lastName, email FROM employees WHERE id = ?";
    RowMapper<Employee> rowMapper = new BeanPropertyRowMapper<Employee>(Employee.class);
    Employee employee = jdbcTemplate.queryForObject(sql, rowMapper, id);
    System.out.println(employee);
    return employee;
    }
    }

在JDBC模板中使用具名参数

具名参数简介

  • 在经典的JDBC语法中,SQL参数是用占位符?表示,并且受到位置的限制。定位参数的问题在于一旦参数的顺序发生变化,就必须改变参数绑定
  • 在Spring JDBC框架中,绑定SQL参数的另一种选择是使用具名参数(named parameter)
  • 具名参数:SQL按名称(以冒号开头)而不是按位置进行指定,具名参数更易于维护,也提升了可读性
  • 具名参数只在NamedParameterJdbcTemplate中得到支持

使用方法

  • Spring配置文件中配置NamedParameterJdbcTemplate

    1
    2
    3
    4
    <!-- 配置NamedParameterJdbcTemplate,该对象可以使用具名参数,其没有无参构造器,必须为其构造器指定参数 -->
    <bean id="namedParameterJdbcTemplate" class="org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate">
    <constructor-arg ref="dataSource"></constructor-arg>
    </bean>
  • 测试

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    @Test
    public void testNamedParameterJdbcTemplate2() {
    String sql = "INSERT INTO employees(last_name, email, dept_id) VALUES(:lastName, :email, :deptId)";
    Employee employee = new Employee();
    employee.setLastName("XYZ");
    employee.setEmail("xyz@gmail.com");
    employee.setDeptId(3);
    SqlParameterSource parameterSource = new BeanPropertySqlParameterSource(employee);
    namedParameterJdbcTemplate.update(sql, parameterSource);
    }

    注意具名参数使用:参数名来表示。

    使用具名参数时可以使用update(String sql, SqlParameterSource paramSource)这个方法来进行更新操作:

    • SQL语句中的参数名和类的属性名一致
    • 使用SqlParameterSource的BeanPropertySqlParameterSource实现类作为参数